# ARDUINO Microcontroller External Interrupt Handling

# User Guide Scalable, Asynchronous Processing of External Interrupts

{A generic method for the handling of multiple external interrupts}

### **Change Control Record**

| Date       | Version   | Remarks                                 | Author      |
|------------|-----------|-----------------------------------------|-------------|
| 03/03/2020 | v2.03.1-D | Based on version 2.03 of the framework. | R D Bentley |
|            |           | Initial version for review              | •           |
|            |           |                                         |             |
|            |           |                                         |             |
|            |           |                                         |             |
|            |           |                                         |             |

Page 2 - 2 -

<sup>©</sup> Copyright of this document is held by the author, R D Bentley. The contents of this document, in part or in whole, may not be reproduced/used for commercial purposes without the express permission of the copyright owner.

## Contents

| A Note About Terminology                                             | 4  |
|----------------------------------------------------------------------|----|
| Introduction                                                         | 5  |
| The Requirements (Objectives)                                        | 6  |
| A Solution (Method)                                                  | 6  |
| Generic Interrupt Handlers                                           | 6  |
| Interrupt Queue                                                      | 7  |
| Free Chain                                                           | 7  |
| Configurable Parameters                                              | 7  |
| Framework Structure                                                  | 8  |
| Diagramatic Conceptual Overview                                      | 8  |
| End User Configurability                                             | 9  |
| C00_Configurations Tab - Diagnostics                                 | 9  |
| C00_Configurations Tab - Interrupt Configuration Data                | 10 |
| Example External Interrupt Parameters                                | 11 |
| C00_Configurations Tab - Interrupt Queue Free Chain Size             | 11 |
| C00_Configurations Tab - max_digital_inputs                          | 12 |
| Performance of Interrupt Queue Handling                              | 12 |
| Global Data and Code Sizes                                           | 13 |
| Sizing Metrics With Diagnostic Code                                  | 13 |
| Sizing Metrics Without Diagnostic Code                               | 13 |
| Free Chain Sizing Influences                                         | 13 |
| Conclusion                                                           | 14 |
| APPENDICES                                                           | 16 |
| Appendix A                                                           | 16 |
| ARDUINO MEGA 2560 External Interrupt to Pin Assignments, by Priority | 16 |
| Appendix B                                                           | 16 |
| ARDUINO pinMode Mode Values                                          | 16 |
| Appendix C                                                           | 16 |
| ARDUINO Interrupt Trigger Values                                     | 16 |
| Appendix D                                                           | 16 |
| ARDUINO Microcontroller Boards with Interrupt Pin Assignments        | 16 |

#### **A Note About Terminology**

- 1. The ARDUINO development environment uses the term 'sketch' to refer to the source C++ code comprising a program. In this guide, rather than 'sketch', the terms 'solution' and 'application' are used as these terms provide a higher level understanding of the intended outcome of an ARDUINO sketch.
- 2. In the context of this guide, the term 'end user' refers specifically to designers and developers of ARDUINO solutions and applications.

Page 4 - 4 -

#### Introduction

This document presents an asynchronous method for a scalable framework for applications requiring the use of external interrupts for ARDUINO microcontrollers (or compatible alternatives, e.g. Elegoo).

The method will allow external interrupts to be processed in their order of priority (when triggered concurrently and in real time), but at such time that the main processing code dictates (asynchronously). That is, the method allows for external interrupts to be captured, as triggered, but actioned outside and away from their associated Interrupt Service Routines (ISRs).

The solution has been developed around the ARDUINO MEGA 2560 microcontroller, but it may be modified for any board in a straight forward way for any number of external interrupts up to the maximum limit set by the specific microcontroller and this framework (see Configuration).

The MEGA 2560 supports up to six external interrupts that are readily accessible to end user developers. These interrupts are mapped to digital input pins; these assignments can be seen at Appendix A. However, the method is easily scaled to use one, two, three,...,or all six external interrupts concurrently, making it an ideal framework on which to base solutions requiring multiple external interrupt processes and processing. For microcontrollers that support more than six interrupts an extension to the underlying framework would be needed. However, it is considered that a limit of six interrupts should not be a limiting factor for most applications likely to be developed. If more are required then the method could be extended by increasing the number of generic ISRs and associated data support structures.

Asynchronous processing is achieved through the design and implementation of an interrupt queue management concept which provides the means to separate the event of an interrupt from its ultimate processing.

When an external interrupt is triggered, it is presented the nominated (generic) ISR mapped and established within the setup() process. ISRs will then do little more than insert the interrupt event into the interrupt queue with no further process taking place at this point. However, to avoid 'bounce' from spurious/dirty interrupts, each ISR incorporates a debounce mechanism which is variable through configuration parameters. Interrupt 'bounce' values may defined differently for each mapped external interrupt, as necessary.

If several external interrupts are triggered at the same time then the ARDUINO microcontroller will 'fire' their respective ISRs in the defined priority order as determined by the microcontroller specifications (see Appendix A for the MEGA 2560 priorities).

The interrupt queue is operated and managed on a first in, first out basis (FIFO). The approach therefore preserves the order in which external interrupts are triggered in the real world.

Following configuration, to meet a specific solution, end users have the ability to add their own code into the 'setup()' and 'main()' segments, as required. The only stipulation here is that, to deal with external interrupts, user added code must utilise a specific 'get interrupt' process (i.e. 'scan IQ()') to progressively action interrupts from the interrupt gueue. This function will either return

1. no interrupts in the interrupt queue, or

2. the value of the generic interrupt handler that was triggered (0, ...,5) (see below for an explanation of generic interrupt handlers). Additionally, a number of variables are set with all of the parameters defined in the configuration data for the respective external interrupt.

There are no other demands placed on the end user solution.

This interrupt framework is intended for use by anyone with an interest in ARDUINO development requiring external interrupts - novices and experts alike. The starting point for its use is to thoroughly understand what it is you want to do with external interrupts. That is, to understand:

- a). why interrupts should be a part of your solution and, if so,
- b). individually, how these should be configured for the solution

It is <u>essential</u> for a successful implementation of this framework that due regard and is time spent analysing the specific requirements for external interrupts and then to configure the framework suitably.

The remainder of this document provides further background and information about the framework, its origins, structure and use.

#### The Requirements (Objectives)

In designing this framework a number of objectives were set, these being to arrive at a solution for external interrupt handling such that:

- disassociates the event of an external interrupt occurring from the processing of its
  occurrence, providing freedom and control to the overlying end user application. That is, a
  solution that would provide independent processing from the 'capture' of received interrupts
  to their eventual processing
- code that could be 'plugged' into without extensive modification (a library would be ideal for this, but for now what is presented is an ARDUINO C++ program framework)
- is scalable, providing a framework to handle multiple external interrupts
- is highly configurable for each possible external interrupt. For example, the need to configure each interrupt differently if needed
- can support unique or nonunique external interrupt event capture/recoding prior to onward processing
- takes account of spurious interrupts. That is, provides debounce for 'dirty' switching.

#### A Solution (Method)

#### **Generic Interrupt Handlers**

By providing a framework that is independent of the specifics of a particular implementation of external interrupt handling it was necessary to introduce the concept of generic interrupt handlers. These generic interrupt handlers are 'agnostic' to any and all interrupts they are configured to service. The allocation of generic interrupt handlers is achieved by the configuration of a data structure that maps the 'real-world' to the virtual. Generic handlers are associated with individual external interrupts and processed according to their associated configured mapped parameters.

This mapping occurs through the table/array 'interrupt\_config\_data' such that each row of this array/table accords with a specific generic interrupt handler (0, ..., 5). For example, row 0 interrupt data is processed by generic\_interrupt0(), row 1 interrupt data is processed by generic\_interrupt1(), etc.

#### **Interrupt Queue**

The method employs the concept of a queue to 'detach' external interrupt events from their onward processing. The queue is managed on a first in, first out basis (FIFO) which ensures that external interrupts are dealt with in the order in which they occur (triggered).

To support this concept, a free chain of data blocks (each of two words) is created in accordance with the size configured (see 'C00\_Configurations Tab - Interrupt Queue Free Chain Size', below).

When an external interrupt is triggered the following process occurs:

- If the external interrupt is defined as being 'unique' then an examination of the interrupt queue occurs to see if the interrupt already has an entry in the queue. If it does not then an entry of the event is inserted into the queue. If it does then it event is ignored
- If the external interrupt is defined as 'nonunique' then and entry is inserted into the queue

#### **Free Chain**

The Interrupt Queue (IQ) is 'fed' from a free chain of blocks that are allocated on demand by the generic interrupt handlers. The size of the free chain is determined by the configuration data value set in the configurations tab (see below C00\_Configurations Tab - Interrupt Queue Free Chain Size').

#### **Configurable Parameters**

The framework is designed to be flexible and readily agreeable to configuration to meet end user external interrupt requirements (within the limits outlined). End user changeable parameters are maintained within one tab, this being C00\_Configurations. See below for a full discussion of end users changeable parameters.

Page 7 - 7 -

#### **Framework Structure**

The section presents an overview of the structure of the framework and how it can be used through configuration.

#### **Diagramatic Conceptual Overview**

At a conceptual level the interrupt framework can be viewed as below:



The framework is developed using ARDUINO IDE (version 1.8.12). The code is structured across a number of tabs to provide logical segmentation. These are:

- 1. A00\_Interrupt\_Framework\_README\_vx.xx
- 2. C00\_Configurations
- 3. E00\_Queue\_Handlers
- 4. E10\_Interrupt\_Handlers
- 5. E90\_Diags
- 6. G00\_Setup
- 7. H00\_Main\_Segment
- 8. T00\_Testing

Page 8 - 8 -

#### **End User Configurability**

For end users, the only tab segments requiring any modification are:

- C00\_Configurations
- E90\_Diags
- G00\_Setup
- H00\_Main\_Segment

The framework is designed such that end user solutions must ensure that the Interrupt Queue is regularly serviced. To achieve this, the framework provides a structure into which an end user code can be added.

End user code (the end user solution) should be inserted at tabs G00-Setup, E90\_Diags and H00\_Main\_Segment. There is no further discussion regarding this, other than to say DO NOT remove or change any existing and resident variables or code in these segments. Simply insert end user code where indicated and/or appropriate.

Configuration of the framework occurs <u>exclusively</u> in the <u>C00\_Configurations</u> tab. The remainder of this section therefore provides an overview of some of the possible ways in which external interrupts and the framework generally may be configured.

There are three possible end user changeable parameters that may be varied according to need. These are:

- Diagnostics
   reporting is on or off
- a #define declaration that toggles if diagnostics
- 2. <u>Interrupt Configuration Data</u> an array/table that describes the specific configuration parameters for each possible external interrupt up to the maximum number of digital interrupt pins possible (for the ARDUINO MEG 2560 this is six)
- 3. <u>Interrupt Queue Free Chain Size</u> a #define declaration that specifies the size of the free chain queue used to store interrupts as they are triggered

These are described below:

#### **C00** Configurations Tab - *Diagnostics*

Parameter definitions:

# Declaration #define bool diags\_on = true;

#### <u>Meaning</u>

the permissible values for this flag are 'true' or 'false'. If set to 'true' (diagnostics on) then the routines predefined in the E90 Diagnostics tab will be actioned if called.

There are two routines available to the end user to aid diagnostics:

- 1. print\_free\_chain() this routine prints the free chain of blocks plus its start of chain pointer.
- 2. print\_IQ() this routine prints the interrupt queue which shows all interrupts in the queue with associated interrupt values.

Note that the above two routines will disable interrupts during their run.

This same flag may be similarly used for any other diagnostic code required by the end user.

#### C00\_Configurations Tab - Interrupt Configuration Data

#### Parameter definitions:

| i alametei delimitoris.                                    |                                                                                                                                                                                                                                                                                                                               |
|------------------------------------------------------------|-------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| <u>Declaration</u>                                         | Meaning                                                                                                                                                                                                                                                                                                                       |
| volatile int interrupt_config_data [max_digital_inputs][7] | This is the principal table (array) controlling how external interrupts are configured. Note that the number of rows parameter ('max_digital_inputs') is preset and should NOT be changed without some extension of existing code. This value is set to the maximum number of interrupts possible on the MEGA 2560 board (6). |
|                                                            | The rows of this table contain the necessary configuration data for each external interrupt to be defined.                                                                                                                                                                                                                    |
|                                                            | Note that the order of the row interrupt data in this array/table is <u>not</u> dependent of the priority of the external interrupts. External interrupt data may be entered in any order, row by row. This is because external interrupts are allocated a                                                                    |

requirement of the framework.

generic interrupt handler during setup(). Ordering external interrupts during configuration is therefore not important or a

| 'interrupt_config_data' column definitions:   |                                                                                                                                                                                                                                                                                                                                                                                               |  |
|-----------------------------------------------|-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|--|
| Declaration Presets Table (Array) Column Data | Meaning The column data of each of row is interpreted identically as follows below:                                                                                                                                                                                                                                                                                                           |  |
| Column 0, active/inactive                     | This value defines if this external interrupt (defined by the row) is active or not. If not used then this should be set to 'inactive'.                                                                                                                                                                                                                                                       |  |
| Column 1, digital pin number                  | This specifies the ARDUINO pin number associated with the external interrupt defined by this row. See ARDUINO data specification for what pins are mapped to what interrupts. For the MEGA 2560 these are detailed at append A.                                                                                                                                                               |  |
| Column 2, pinMode mode value                  | During setup(), each 'active' external interrupt is defined via a pinMode call. This call uses the pinMode mode value defined by this value. See Appendix B for mode types.                                                                                                                                                                                                                   |  |
| Column 3, unique/nonunique flag               | If set to 'true' then the interrupt handlers will not insert an external interrupt triggered event into the interrupt queue if one already exists (unique). If set to 'false' then multiple external interrupt trigger events may exist in the interrupt queue (nonunique). Clearly, each interrupt can be set to either value allowing a mix on unique and nonunique processing if required. |  |
| Column 4, trigger type                        | External interrupts are linked to digital pins and ISRs during the setup() process. At this time, the attach interrupt call will use the value held in this column to determine the type of trigger required. See Appendix C for trigger types.                                                                                                                                               |  |
| Column 5, debounce value                      | When external interrupts trigger it is often the case that spurious interrupts are also raised by the physical event that led to the interrupt. This parameter ('debounce') provides a window during which external interrupts arriving are ignored.                                                                                                                                          |  |

| <u>Declaration Presets</u>   | Meaning The value defined here is in milliseconds (msecs).                                  |
|------------------------------|---------------------------------------------------------------------------------------------|
| Column 6, external interrupt | This defines the external interrupt number that is linked to defined digital interrupt pin. |

#### Example External Interrupt Parameters

```
volatile int interrupt config data[max digital inputs1][7]=
// 0
          1
                                        4
                     2
                                3
                                                  6 comment
               INPUT PULLUP,
          21,
                                      true,
                                             15, 0 // generic interrupt
active,
                             RISING,
                                                    handler 0 entries
               INPUT PULLUP,
                             RISING, false, 5, 1 // generic interrupt
          20,
active,
                                                    handler 1 entries
                                     true, 15, 2 // generic interrupt
active,
          19,
               INPUT PULLUP,
                             RISING,
                                                    handler 2 entries
                                     true, 15, 3 // generic interrupt
inactive, 18,
               INPUT PULLUP,
                             RISING,
                                                    handler 3 entries
         2,
               INPUT PULLUP,
                             RISING, false, 5, 4 // generic interrupt
active,
                                                    handler 4 entries
active, 3,
               INPUT PULLUP,
                             RISING,
                                      true, 15, 5 // generic interrupt
                                                     handler 5 entries
```

#### C00\_Configurations Tab - Interrupt Queue Free Chain Size

A free chain of data blocks (2 words each) is created during the set up process, the number of blocks being defined by the variable 'max IQ free chain blocks'. Initially, the free chain is configured in accordance with the value defined by the declaration:

#### Parameter definitions:

| <u>Declaration</u>       |  |  |
|--------------------------|--|--|
| #define                  |  |  |
| max IQ free chain blocks |  |  |
| 64                       |  |  |

#### Meaning

The free chain will be created with this number of entries (blocks). Each free chain block comprises 2 words (integers). The first word of each free block contains a forward pointer to the next free block or -1 (end of chain value) if the last free block.

In this example, the free chain would be created as 64 available blocks to be allocated to the Interrupt Queue (IQ) on demand.

However, it can also be seen that some interrupts in the above example are defined to be nonunique. That is, more than one interrupt event can be in the IQ at the same time.

If nonunique interrupts are required then the value of this #definition must be carefully considered, otherwise, triggered interrupts may not be able to be inserted into the interrupt queue if it is too small.

Selecting a value may not be straight forward and trial and error may be needed dependent on the specifics of the

Page 11 - 11 -

max\_digital\_inputs - this is set to 6 for this framework, reflecting the number of external interrupts supported by the MEGA 2560

| <u>Declaration</u> | <u>Meaning</u>                                                  |
|--------------------|-----------------------------------------------------------------|
|                    | application being developed. However, unless 100's of           |
|                    | interrupts are being generate per second then you may find      |
|                    | that a value of, say, 16, 32, 64 or 128 would be of adequate    |
|                    | size for the free chain. Note that it is not a requirement that |
|                    | free chain size is a power of 2. It can be any value, e.g. 17,  |
|                    | 29, etc.                                                        |

#### C00 Configurations Tab - max digital inputs

There is an additional parameter defined within the C00 Configurations tab, but this is not for end user modification. It is mentioned here for completeness, only.

| Parameter definitions:                  |                                                                                                                                                                                                                                                           |
|-----------------------------------------|-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| <u>Declaration</u>                      | Meaning                                                                                                                                                                                                                                                   |
| <pre>#define max_digital_inputs 6</pre> | This declaration is included here for completeness – it is <u>not</u> an end user changeable parameter.                                                                                                                                                   |
|                                         | This declaration defines the number of external interrupts that the framework is designed for - this reflects the number of generic interrupt handlers defined.                                                                                           |
|                                         | For the current version of the framework this is the maximum number of external interrupts available on the ARDUINO MEGA 2560 microcontroller, hence it is six. See Appendix A for a list of the MEGA 2560 digital pins that support external interrupts. |

#### **Performance of Interrupt Queue Handling**

It goes without saying that dealing with interrupts via way of an intermediary gueuing system must come with some overhead baggage, this being in storage space and execution time. Where an application is time critical in dealing with external interrupts then it may be better to design the solution directly such that ISRs form the core of the construction.

However, to aide this judgement, some performance measurements have been taken to understand the timings for interrupt queue management. Testing examined:

- a. inserting interrupts into the interrupt queue until it was fully saturated (fully loaded),
- b. removing interrupts successively from the interrupt queue until it was fully emptied.

Tests were run for different sizes of interrupt queues – 1, 2, 4, 8, 16, 32, 64, 128, 256, 512, 1024<sup>2</sup> queue entries. What was found was that timings were largely not at all affected by changing the size of the free chain/interrupt queue. To ensure optimal processing, the interrupt debounce value was set to 0 milliseconds, which may not be real world justified for some uses, and with unique interrupt processing turned off (allowing any number of the same interrupts in the interrupt queue at the same time).

The results obtained showed that:

- 1. the average time to insert an interrupt into the interrupt queue is 15 microseconds,
- 2. the average time to remove an interrupt from the interrupt queue is 12 microseconds.

There is nothing special about this size progression. The queue size can be any value.

Clearly, a little less work taking out from the queues than putting in!

#### Global Data and Code Sizes

Measurement were taken to determine the respective sizes of global data and framework code, with and without diagnostic code included.

The only variable data impacting on global data allocation relates to the size of the free chain initialised for the interrupt queue. The base value taken for determining sizing was 16 free chain blocks (i.e. 64 bytes of data requirement). This provided the following results:

#### **Sizing Metrics With Diagnostic Code**

Total allocation size for global data including the free chain: 409 bytes (this value includes the

free chain sized at 16 blocks, each of 4 bytes (total 64 bytes).)

Total size of generated code, excluding global data: 5,392 bytes Total size (data and code) 5,801 bytes

#### **Sizing Metrics Without Diagnostic Code**

Total allocation size for global data including the free chain: 234 bytes (this value includes the

free chain sized at 16 blocks, each of 4 bytes (total 64 bytes).)

Total size of generated code, excluding global data: 4,346 bytes Total size (data and code) 4,580 bytes

#### Free Chain Sizing Influences

Clearly, the greater the size of the free chain the more global memory will be allocated. However, the relationship between the number of free chain blocks and additional memory requirements is completely linear. That is, using sizing data provided above, then if a free chain size of 32 blocks is required then the respective additional global data size increase by 4 x 16, i.e. 64 bytes, as the sizing above already includes 16 free chain blocks. The logic is similarly applied for any other free chain size. It is simply:

Additional global data space = (Required size (in blocks) –  $16^3$ ) x 4 bytes

For each additional free chain block increase above the initial 16 configured for the sizing model the global data requirement will increase by 4 bytes.

Using these metrics it will therefore be possible to determine how global space will be allocated just for the framework alone, this being a function purely of the free chain size.

<sup>16</sup> free chain blocks already included in based sizing model

#### Conclusion

The development of the interrupt framework was steered by the objectives established at the outset. The final product has met these objectives as follows:

#### **Objective**

# disassociates the event of an interrupt occurring from the processing of its occurrence, providing freedom and control to the overlying end user application. That is, a solution that would provide independent processing from the 'capture' of received interrupts to their eventual processing.

#### **How Met**

Object met. The solution is based on the concept of a queue of interrupts that can be processed after an interrupt event and at a time dictated by the end user application.

code that could be 'plugged' into without extensive modification.

Objective met. End user coding occurs in the setup() and main() segments. Additional segments may be added/included as necessary without change to any framework element.

is scalable, providing a framework to handle multiple external interrupts

Objective met. The framework will handle from one to six concurrent interrupts, this being constrained by a design based on the MEGA 2560 microcontroller. Further interrupt handling can be provided by suitable extension. The solution is therefore extensible.

is highly configurable for each possible interrupt. For example, the need to configure each interrupt differently if needed

Objective met. Interrupt handling is highly configurable depending on end user requirements. Each of the six external interrupt definitions can be uniquely configured in required.

can support unique or nonunique interrupt event capture/recoding prior to onward processing

Objective met. Interrupt handling is designed to allow external interrupt events to be recorded as 'unique' or 'nonunique' and suitably recorded within the interrupt queue.

takes account of spurious interrupts. That is, provides debounce for 'dirty' switching.

Objective met. Generic interrupt handlers are coded to take account of 'interrupt bounce' as may occur, for example, from a 'dirty' switch. Debounce can be configured for each external interrupt in milliseconds (msecs). If set to 0 msecs, then all interrupt events will be record, unless configured as 'unique'.

Note also that the debounce code takes regard of the millis() function wrap around when it overflows back to 0 msecs.

Whether the framework proves to be helpful in implementation will depend on the specific needs of the end user solution sought for the capture and processing of external interrupts. Although the performance data gathered (see above) suggests that the method is, on the

# ARDUINO Interrupt Handling Scalable, Asynchronous Processing of External Interrupts

whole, efficient and provides good levels of performance, it use must be judged as to whether it is applicable for the need.

If absolute performance is the principal requirement then perhaps a directly crafted solution is appropriate, one with no/few overheads. However, it is considered that for most every day requirements the framework will provide a very helpful basis on which to deal with external interrupts in an application.

#### **APPENDICES**

#### **Appendix A**

ARDUINO MEGA 2560 External Interrupt to Pin Assignments, by Priority

| External Interrupt Number (by priority) | Pin Assignment |
|-----------------------------------------|----------------|
| INT0                                    | 21             |
| INT1                                    | 20             |
| INT2                                    | 19             |
| INT3                                    | 18             |
| INT4                                    | 2              |
| INT5                                    | 3              |

#### **Appendix B**

#### **ARDUINO pinMode Mode Values**

| Mode Values  |
|--------------|
| INPUT        |
| INPUT_PULLUP |
| OUTPUT       |

#### **Appendix C**

#### **ARDUINO Interrupt Trigger Values**

| Trigger Values |  |  |
|----------------|--|--|
| LOW            |  |  |
| CHANGE         |  |  |
| FALLING        |  |  |
| RISING         |  |  |
| HIGH⁴          |  |  |

#### **Appendix D**

#### **ARDUINO Microcontroller Boards with Interrupt Pin Assignments**

| Microcontroller Board             | Digital Pins Usable for Interrupts                                       |
|-----------------------------------|--------------------------------------------------------------------------|
| Uno, Nano, Mini, other 328-based  | 2, 3                                                                     |
| Uno WiFi Rev.2                    | all digital pins                                                         |
| Mega, Mega2560, MegaADK           | 2, 3, 18, 19, 20, 21                                                     |
| Micro, Leonardo, other 32u4-based | 0, 1, 2, 3, 7                                                            |
| Zero                              | all digital pins, except 4                                               |
| MKR Family boards                 | 0, 1, 4, 5, 6, 7, 8, 9, A1, A2                                           |
| Due                               | all digital pins                                                         |
| 101                               | all digital pins (only pins 2, 5, 7, 8, 10, 11, 12, 13 work with CHANGE) |

Due, Zero and MKR1000 boards only